# Lab 4: Now You See It, Now You Don't

### **Introduction:**

In this lab, we are going to utilize the VGA analog video standard in order to produce a static image on a display. The timing signals and rom addressing will all be driven by a combination of counters, which prove to be one of the most versatile components in digital design.

### **Prelab – Wires with Electron Guns**

### Background:

In order to understand VGA graphics, one must first understand the hardware that it was intended to be used with. Back in the days of Cathode Ray Tube monitors (CRT), generating a color image was a relatively simple process. The front of the monitor consisted of a plate of glass that had a coating of phosphorescent particles on the back. By firing an electron beam at the particles, they would become energized and emit light. The color of the light would depend on the chemical composition of the particle, and the intensity on the level of energy imparted by the beam. In order to direct the beam on the screen, a set of electromagnetic coils are used to deflect it to various angles on the x and y axes.

In this way, the question then becomes how we can generate multiple colors using this technology. The answer is simple; three types of particles that glow red, green, and blue. By creating combinations of these three colors and exploiting some glitches in the human visual system, we can create a large number of colors.



Figure 1: CRT Monitor Hardware

Obviously, the CRT has gone the way of the Dodo bird but the VGA standard has remained as the lowest common denominator. Almost all displays support the standard and it has only recently begun to be slowly phased out. While it is old and has been superseded by myriad options, it is still a very useful method for getting images out of an FPGA. Moreover, the techniques and theory behind video generation in the context of VGA are the foundation of the other methods (DVI and HDMI for 640x480 simply encode the VGA RGB components using TDMS and serialize them out at 250MHz using differential signals with an explicit clock).

Knowing all about the technology is wonderful, but if it is not going to be used to do something meaningful then it is all for naught. In order to do so, we must first prepare something to display. For this purpose, a simple Matlab script has been created. It takes an input image and downsamples it to an 8 bit color space while also creating a Xilinx COE file that we will use later to create a ROM to hold our image data.

In order to use the script, please download an image of your choosing in JPG format and save it to a folder. When the script is run, it will ask for you to enter the absolute path to that folder (for example, my desktop would be C:\Users\Gregory\Desktop). It will then ask for you to enter the name of the output file you wish to create. It will then take the first JPG in that folder and convert it. The output file will be stored in the same folder as the image that was used for input. As an aside, the script is designed to be tolerant of all resolutions and targets a 480x480 output resolution due to memory limitations on the Zybo board. If images are too big, they will be cropped automatically. If they are too small, they will be padded with black.

#### Prelab Task(s):

- Read section 11 on VGA in the Digilent reference manual for the Zybo board: https://reference.digilentinc.com/reference/programmable-logic/zybo/reference-manual
- Run the provided Matlab script with an image of your choosing. Keep the output file for use later in the lab

### **Part 1 – Timing is Everything:**

### Background:

Now that you understand how the technology for the monitors worked back in the day, you have some preliminary idea about the signals that were used to control those monitors. In the end, VGA can be reduced to 5 signals. Keep in mind that for modern monitors, they interpret these signals differently (they don't use electron guns any more) but the principles are the same.

- R: An analog intensity for the "red" electron gun
- G: An analog intensity for the "green" electron gun
- B: An analog intensity for the "blue" electron gun
- HS (Horizontal Sync): A signal that initiates the "newline" for the electron guns
- VS (Vertical Sync): A signal that initiates the "new page" for the electron guns

Through these 5 signals (and no explicit clock, as it is recovered by the monitor via doing some math on the HS and VS signals), we output color intensities to a continuously moving set of electron beams.

While the R, G, and B signals are relatively obvious in what they are, the same cannot be said for HS and VS. In order to understand those, we have to look at several tables and diagrams.

The first thing that we realize when we look at these diagrams is that even though our screen resolution is 640x480, we actually base our signals off an 800x525 "screen". This is because there is off screen space both below and to the right of the image. In the old days, this existed to allow the electron guns time to move over when we told them to go the next line or next screen.



Figure 2: Off Screen Space and Sync Pulses Visualization

Now that we understand why the sync pulses (HS and VS) exist, and that they exist off screen, the question then becomes when they are asserted. In order to find that out, we can look at these two handy tables.

# Horizontal timing (line)

Polarity of horizontal sync pulse is negative.

| Scanline part | Pixels | Time [µs]        |
|---------------|--------|------------------|
| Visible area  | 640    | 25.422045680238  |
| Front porch   | 16     | 0.63555114200596 |
| Sync pulse    | 96     | 3.8133068520357  |
| Back porch    | 48     | 1.9066534260179  |
| Whole line    | 800    | 31.777557100298  |

Figure 3: Horizontal Timing Table

# Vertical timing (frame)

Polarity of vertical sync pulse is negative.

| Frame part   | Lines | Time [ms]         |
|--------------|-------|-------------------|
| Visible area | 480   | 15.253227408143   |
| Front porch  | 10    | 0.31777557100298  |
| Sync pulse   | 2     | 0.063555114200596 |
| Back porch   | 33    | 1.0486593843098   |
| Whole frame  | 525   | 16.683217477656   |

Figure 4: Vertical Timing Table

\*Note: The porches just mean space before and after the sync pulse during which the output is held black (R=G=B=0)

From these tables, and the info above, we can ascertain the following:

- Our timing generator is composed of two counters:
  - A horizontal counter that counts from [0, 799]. It increments every clock tick when enable is '1'.
  - A vertical counter that counts from [0, 524]. It increments every clock tick when enable is '1' and the horizontal counter has reset to 0.
- During the time that our Horizontal counter is between [0, 639], and our Vertical counter is between [0, 479], our display is ON and we output color data. Otherwise we output black (R=G=B=0).
- During the time that our Horizontal counter is between [656, 751], our HS signal is '0'. Otherwise it is '1'
- During the time that our Vertical counter is between [490, 491], our VS signal is '0'. Otherwise it is '1'

#### Task(s):

- Create an entity called "vga\_ctrl" that takes as input a clock and a clock enable and produces the following outputs. It should behave according to the description above.
  - o "hcount": a 10 bit std logic vector that is the value of the horizontal counter
  - o "vcount": a 10 bit std\_logic\_vector that is the value of the vertical counter
  - o "vid": a 1-bit signal that is 1 when the display should be on, otherwise it is '0'
  - o "hs": a 1-bit signal that is the HS pulse
  - o "vs": a 1-bit signal that is the VS pulse
- Run some basic simulation tests to check the behavior of the counters and the hs and vid signals. Do not worry about the vs signal, your simulation would have to go a very long time to witness changes on it.

### Part 2 – Pixel Pusher:

#### Background:

In order to read our picture onto the screen, we need to store the pixel information into a memory block. The easiest way to do so is using the Xilinx Block Memory IP, which using hard memory elements embedded in the FPGA fabric.

In order to do so, click "IP Catalog" under the Project Manager tab. Then enter "block" into the search bar, and double click the "Block Memory Generator" IP to start the customization process.

Change the Component Name to "picture", and the Memory Type to "Single Port ROM". Then navigate to the "Port A Options" tab. Change the width to 8 and the depth to 230400. Change the Enable Port Type to "Always Enabled" and Un-tick any output register boxes. In the "Other Options" tab, tick the "Load Init File" box and use the browse button to direct it to the COE file generated during the prelab. Then click the OK button. The tool will bring up a window asking for permission to generate the required files for the IP instance. Click the Generate button. Wait patiently; the process may take quite some time (around 10 minutes) depending on the speed of your machine.

Once Vivado has finished generating the IP, expand it in the project manager tab with the drop down arrow and open the main VHDL file. Using the Entity declaration found in the VHDL file, we can later instantiate it as a component in our top level design.

#### Task(s):

- Create a block memory according to the description above with the COE file generated during the prelab section
- Create an entity called "pixel\_pusher" that takes as input a clock, a clock enable, a 1-bit "VS", an 8-bit "pixel" signal, a 10-bit "hount" signal, and a "vid" signal. It should output two 5-bit "R" and "B" signals and a 6-bit "G" signal, as well as an 18-bit "addr". It should have the following behavior:
  - o It contains an internal 18 bit counter called addr with the following behavior:
    - Every clock tick when enable is '1', vid is '1', and hount is less than 480, it increments. It resets synchronously when VS is '0'.
  - o Every clock tick when enable is '1', vid is '1', and hount is less than 480
    - R <= pixel(7 downto 5) & "00"
    - G <= pixel(4 downto 2) & "000"
    - B <= pixel(1 downto 0) & "000"

Otherwise, R, G, and B are 0

• Draw a block diagram for a top level design that contains instances of your block memory, "pixel\_pusher", "vga\_ctrl", and a "clock\_div" modified to produce a 25 MHz clock enable. It should take as input a clk signal and produce as output single bit vga\_hs and vga\_vs signals, 5-bit vga\_r and vga\_b signals, and a 6-bit vga\_g signal

- Create a top level entity called "image top" that implements your block diagram
- Create an appropriately modified XDC file and load the design onto the board to test it

Here is a simplified system block diagram to help with the creation of the top level design. Please flesh it out with all of the proper signal wires and bit widths for your drawn version.



Figure 5: Simplified System Block Diagram

## **Testing:**

Once your design has been loaded onto the board, you will need to connect a VGA cable from the port on the Zybo device to a computer monitor. Because the monitors in EE103 do not have VGA inputs, we will be using a mix of the monitors in EE105 and some other ones that are not part of the normal lab setup.

When your board is connected, you should see your image used during the prelab Matlab conversion displayed on the screen, positioned at the top left area of the monitor.

# **Report:**

See the associated lab report format guideline in Resources

### **Sources:**

https://www.xilinx.com/support/documentation/data\_sheets/ds190-Zynq-7000-Overview.pdf

http://tinyvga.com/vga-timing/640x480@60Hz

https://i.stack.imgur.com/jURiy.jpg

https://reference.digilentinc.com/reference/programmable-logic/zybo/reference-manual